/*日志落地模块的实现
    1.抽象落地基类
    2.派生子类(根据不同的落地方向进行派生)
    3.使用工厂模式进行创建与表示的分离
*/

#ifndef __M_SINK_H__
#define __M_SINK_H__

#include"util.hpp"
#include<memory>
#include<fstream>
#include<sstream>
#include<cassert>

namespace bitlog{
    class LogSink{
        public:
            using ptr = std::shared_ptr<LogSink>;
            LogSink() {}
            virtual ~LogSink(){}
            virtual void log(const char *data,size_t len)=0;
    };
    //落地方向： 标准输出
    class StdoutSink:public LogSink{
        public:
        //将日志消息写入到标准输出
        void log(const char*data,size_t len)
        {
            std::cout.write(data,len);
        }
    };

    //落地方向： 指定文件
    class FileSink:public LogSink{
        public:
        //构造时传入文件名，并打开文件，将操作句柄管理起来
        FileSink(const std::string&pathname):_pathname(pathname)
        {
            //1.创建日志文件所在的目录
            util::File::createDirectory(util::File::path(pathname));
            //2.创建并打开日志文件
            _ofs.open(_pathname,std::ios::binary|std::ios::app);
            assert(_ofs.is_open());
        }
        
        void log(const char *data,size_t len)
        {
            _ofs.write(data,len);
            assert(_ofs.good());
        }
        private:
            std::string _pathname;
            std::ofstream _ofs;        //输出的时候可以直接通过句柄进行输出，不用每次先写入消息打开文件和关闭文件
    };
    //落地方向： 滚动文件(以大小进行滚动)
    class RollBySizeSink:public LogSink{
        public:
        //构造时传入文件名，并打开文件，将操作句柄管理起来
        RollBySizeSink(const std::string &basename,size_t max_size):
        _basename(basename),_max_fsize(max_size),_cur_fsize(0),_name_count(0)
        {
            std::string pathname=createNewFile();
             //1.创建日志文件所在的目录
            util::File::createDirectory(util::File::path(pathname));
            //2.创建并打开日志文件
            _ofs.open(pathname,std::ios::binary|std::ios::app);
            assert(_ofs.is_open());
        }
        //将日志消息写入到标准输出--写入前判断文件大小，超过了最大大小就要切换文件
        void log(const char *data,size_t len)
        {
            if(_cur_fsize>=_max_fsize)
            {
                 _ofs.close(); //关闭原来已经打开的文件
                 std::string pathname =createNewFile();
                 _ofs.open(pathname,std::ios::binary|std::ios::app);
                 assert(_ofs.is_open());
                 _cur_fsize=0;
            }
            _ofs.write(data,len);
            assert(_ofs.good());
            _cur_fsize+=len;
        }

        private:
            std::string createNewFile()//进行大小判断，超过指定大小则创建文件
            {
                //获取系统时间，以时间来构造文件名扩展名
                time_t t=util::Date::now();
                struct tm lt;
                localtime_r(&t,&lt);
                std::stringstream filename;
                filename << _basename;
                filename << lt.tm_year+1900;
                filename <<lt.tm_mon+1;
                filename<<lt.tm_mday;
                filename << lt.tm_hour;
                filename <<lt.tm_min;
                filename<<lt.tm_sec;
                filename<<"-";
                filename<<_name_count++;
                filename<<".log";    //文件后缀名
                return filename.str();
            }

        private:
            //基础文件名通过基础文件名+扩展文件名(以时间生成)组成一个实际的当前输出文件名
            size_t _name_count;
            std::string _basename;//  ./logs/base-       ->./logs/base-20002008091231.log
            std::ofstream _ofs;  
            size_t _max_fsize;  //记录最大大小，当前文件超过了这个大小就要切换文件
            size_t _cur_fsize;//记录当前文件已经写入的数据大小
    };

   
    class SinkFactory{
        public:
           template<typename SinkType,typename ...Args>
            static LogSink::ptr create(Args && ...args)
            {
                return std::make_shared<SinkType>(std::forward<Args>(args)...);
            }
    };
}

#endif